Introduction

Motivation

Indego is the de facto bike sharing network in Philadelphia. It has become a popular, more eco-friendly alternative for short trips across the city since launching in late April of 2015. However, little has been done to analyze the publicly available data associated with Indego. Thus, we think it would be beneficial to the public if we quantify and assess several pertinent features and patterns related to Indego, generating insights on how the bike sharing network is utilized across the city. More specifically, we want to more closely study and answer questions related to the following four themes:

  • Station usage (usage rate, peak periods, popular rental/return stations, etc.)
  • Daily traffic (popular routes, volume across different sections, travel patterns, etc.)
  • Trip durations (length of trips, median rental/return time, etc.)
  • User profiles (type of passholders, frequency of round-trips, seasonality, etc.)

Each of the above four features will be explained in finer details in subsequent sections. To realize this objective, we have created static plots (shown in this notebook) to describe salient themes as well as a Shiny app (Github), which also contains real-time status updates of all Indego stations across Philadelphia. We will also provide instructions on runnning our app further down in our report.

Our objective starting out was to examine bike share programs in the most densely populated cities across the eastern United States, mainly via exploratory data analysis and visualization techinques. We initially considered NYC’s Citibike as a promising candidate, but it has already been extensively analyzed (see Todd Schneider’s blogpost for an excellent example). The marginal returns of analyzing such a dataset are thus minimal, if not, nil. As a result, we decided on Philadelphia, primarliy for two reasons: it is the second largest city satisfying our predefined criteria (after NYC), and also has a relatively new bike share program. Thus, we thought the Indego dataset would be an ideal choice. It would also be interesting to use our dataset to compare and contrast New York and Philadelphia’s bike share programs.

Data Source

The dataset is publicly available on Indego’s website. It can be found by clicking the About panel and then selecting Data. All quarterly data are listed under the section The Data, in which each station’s information is available through the Station Table hyperlink. In addition, real-time data used in our Shiny app is available under the section Indego Station Status via the GeoJSON hyperlink.

Team

  1. Kejia Shi (ks3403): data processing, exploratory data analysis, visualization (static and Shiny plots), writing/editing report.
  2. Mason Sun (zs2321): data processing, exploratory data analysis, developing front/backend of Shiny app, writing/editing report.
  3. Hung-Shi Lin (hl2997): proofread.

Analysis of Data Quality

Quarterly data

For quarterly data, we had seven files in total, each representing data between Q2 of 2015 and Q4 of 2016, inclusively. Each contained the same columns:

  • trip_id (unique identifier of a bike trip)
  • duration (length of trip in seconds or minutes calculated by check-in and check-out time)
  • start_time (clock start time and calendar date)
  • end_time (clock end time and calenar date)
  • start_station_id (unique identifier of starting station)
  • start_lat (latitude of starting station)
  • start_lon (longitude of starting station)
  • end_station_id (unique identifier of end station)
  • end_lat (latitude of end station)
  • end_lon (longitude of end station)
  • bike_id (unique identifier of the bike used during the trip)
  • plan_duration (type of pass by number of days, e.g. thirty-day pass is shown as 30)
  • trip_route_category (round trip or one-way)
  • passholder_type (walkup, IndegoFlex, or Indego30)

We combined these into one data frame (called df) using the code as follows. The description of our preprocessing methods are embedded as comments in the code chunk below. Please see them for more detailed explanations.

library(tidyverse)
### read in quarterly data ###
Y16Q4 <- read.csv("data/Indego_Trips_2016Q4.csv", header=T)
Y16Q3 <- read.csv("data/Indego_Trips_2016Q3.csv", header=T)
Y16Q2 <- read.csv("data/Indego_Trips_2016Q2.csv", header=T)
Y16Q1 <- read.csv("data/Indego_Trips_2016Q1.csv", header=T)
Y15Q4 <- read.csv("data/Indego_Trips_2015Q4.csv", header=T)
Y15Q3 <- read.csv("data/Indego_Trips_2015Q3.csv", header=T)
Y15Q2 <- read.csv("data/Indego_Trips_2015Q2.csv", header=T)
indego_stations <- read.csv("data/indego_stations.csv", header=T)
### preprocessing ###
# select currently active stations from most recent quarterly dataset
station_loc <- Y16Q4 %>% dplyr::select(start_station_id, start_lat, start_lon) %>% unique
# create a dataframe that includes the relevant columns for all active stations by left joining on the active stations and all stations, ignoring any rows that contain missing values
station_loc <- dplyr::left_join(station_loc, indego_stations, by = c("start_station_id" = "Station.ID")) %>% na.omit %>% .[, -6]
# rename columns of dataframe
colnames(station_loc) <- c("id", "lat", "long", "name", "date")
# remove whitespaces in the date column
station_loc$date = trim(station_loc$date %>% as.character())

# below are two auxiliary fuction to parse the start/end time and date of each trip, which displays them as a single string. We partition them into separate columns that now display year, month, day, hour, minutes, and date (YYYY-MM-DD).

# for quarters of 2015
parseTime_15 <- function(df) {
  df$s_hour <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%H")
  df$s_min <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%M")
  df$e_hour <- format(as.POSIXct(strptime(df$end_time,"%m/%d/%y %H:%M",tz="")) ,format = "%H")
  df$e_min <- format(as.POSIXct(strptime(df$end_time,"%m/%d/%y %H:%M",tz="")) ,format = "%M")
  df$year <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%Y")
  df$month <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%m")
  df$day <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%d")
  df$date <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%Y-%m-%d")
  return(df)
}

# for quarters of 2016, which are defined differently from 2015
parseTime_16 <- function(df) {
  df$s_hour <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%H")
  df$s_min <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%M")
  df$e_hour <- format(as.POSIXct(strptime(df$end_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%H")
  df$e_min <- format(as.POSIXct(strptime(df$end_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%M")
  df$year <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%Y")
  df$month <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%m")
  df$day <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%d")
  df$date <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%Y-%m-%d")
  return(df)
}

# pass into respective auxiliary functinos
Y15Q2 <- parseTime_15(Y15Q2)
Y15Q3 <- parseTime_15(Y15Q3)
Y15Q4 <- parseTime_15(Y15Q4)
Y16Q1 <- parseTime_16(Y16Q1)
Y16Q2 <- parseTime_16(Y16Q2)
Y16Q3 <- parseTime_16(Y16Q3)
Y16Q4 <- parseTime_16(Y16Q4)

# combine all quarterly data into a single data frame called df
df <- rbind(Y15Q2, Y15Q3, Y15Q4, Y16Q1, Y16Q2, Y16Q3, Y16Q4)

# standardize duration into units of seconds
# also removes columns not needed in subsequent analyses
dataClean <- function(df) {
  df$duration <- df$duration / 60
  df[, -c(1, 3, 4, 6, 7, 9, 10, 12)]
}
df <- dataClean(df)

# remove individual dataframes
rm(Y15Q2, Y15Q3, Y15Q4, Y16Q1, Y16Q2, Y16Q3, Y16Q4)

During our initial exploration of the dataset, we realized some of the respective datasets for the quarterly data contained missing values, which would imply that our final dataframe also contains missing values. We thus try to qualitatively visualize these features, show below:

# ignore columns after passholder_type (col_num = 14) as subsequent columns are defined by us.
library(extracat)
visna(df[,c(1:14)])

missing_table = table(is.na(df))
cat("Percentage of missing entries:", missing_table[2]/sum(missing_table)*100)
Percentage of missing entries: 0.01720519

We see that the proportion of missing values is miniscule relative to our whole dataset (approximately 0.0172%). Moreover, they are all related to latitude and longitude data as well as the bike ID. If need be, we can infer the latitude and longitude from the station name or station ID, but we actively chose to ignore rows corresponding to these missing values since we cannot explicate why these values are missing. We thus choose to ignore them altogether.

Indego stations data

For the data detailing the information of each station (indego_stations.csv), we had to perform a bit of data preprocessing to get it into an ideal format. The data contained the columns as follows:

  • Station ID (unique identifier of a Indego station)
  • Station Name (name of corresponding Indego station, usually a street address)
  • Go live date (the date that the station started operations)
  • Status (its current status, either active or inactive)

There were many leading and trailing whitespaces in the Station Name column, so performed the following to remove these unneeded characters:

# load stations data
indego_stations <- read.csv("../data/indego_stations.csv", header=T)
# a function that removes leading and trailing white spaces of a given string
trim <- function (x) gsub("^\\s+|\\s+$", "", x)
# remove whitespaces surrounding each entry in the column of Station.Name
indego_stations$Station.Name <- trim(indego_stations$Station.Name)

There were no missing values in this table, so we do not provide plots related to this here.

GeoJSON

For real-time status updates of Indego stations, we use the semi-structured GeoJSON dataset provided by Indego. We allow the user to manually refresh our Shiny app, in which we call the GeoJSON file in real-time to get a visualization update (more specific details in subsequent sections). We used the R packpage geojsonio to unpack and transform this data into a workable format as follows:

library(geojsonio)
station_status <- fromJSON("https://gbfs.bcycle.com/bcycle_indego/station_status.json")

Executive Summary

The primary objective of this report is to derive valuable insights of deploying shared bikes for big cities (in our case, Philadelphia’s Indego), and how they are utilized by the public mass. More specifically, we want to understand patterns of usage among different stations and users over time. The exploratory methodologies that we implement on the Indego dataset will generally follow conventional principles of analyzing temporal and spatial features of large datasets.

Upon retrieving the dataset, we immediately raised the following questions:

  1. For the total 105 stations deployed gradually within two years, which stations are the most popular ones? Has the popularity changed over time?

  2. For specific stations we have an expection on their daily rental patterns. How do we visualize and examine such patterns?

  3. With the geographical data of stations and GeoSON file, how do we draw connections among stations and see rental and return patterns, popular routes, and visualize the real-time station status?

We present here a quick run-through of the plots we present in our main analysis, where we will dive into much finer details. The following plots demonstrate the main findings of our exploratory data analysis.

Firstly, we make general line plots with monthly frequency data and further separate the data into week days and weekends. From the plots, we see clearly the seasonal usage variation.

Secondly, we pick three different stations located at the university area, downtown and suburban Philadelphia, and draw the average hourly trip durations for week days and weekends in a selected time period. Travel patterns among different types of stations thus becomes evident. For example, the weekends’ average trip duration in the afternoon is longer than that of the weekday’s for users close to university campuses.

Thirdly, we utilize leaflet to create interactive maps in order to represent the spatial information we can derive from the dataset. We effectively represented the average daily rental and return frequencies among stations using different dot sizes (larger dots implies higher frequency). We embed the real-time GeoJSON file into our Shiny app to show real-time updates of station status. In order to jointly analyze the spatial and time dimensions, we pick the ten most popular rental and return stations for two recent quarters to determine differences in trends. Besides the plots below, we also made a map for the five most popular routes (see next part).

Moreover, we analyze the daily trip durations and the user profiles, and uncover the the shift of passholder types and trip route (one-way or round-trip) changes. The most incisive observation we got from this section is that the percentage of the most popular passholder type (Indego30) and one-way trips has gotten more popular. In addition, less people are taking round trips.

Lastly, we create a Shiny dashboard to let people explore the dataset as their wishes. That is, they can select stations and date ranges of interest, and also modify parameters of different charts (e.g. number of bins for histograms) to generate customized insights.

Main analysis

In this main section, we present our findings of different facets pertaining to the Indego dataset. We subdivide our analysis into four interrelated sections: station usage, daily traffic, trip duration, and user profiles, in that particualr order. We will walk the reader through our thought process and findings, and finally present some concluding remarks.

Station usage

In this section, we examine the usage rate among the different Indego stations across Philadelphia. We motivate our analysis from a holistic perspective towards more granular analysis. This structural layout also holds true for subsequent sections. With that in mind, let us first patterns related to some of the more popular Indego stations. We subdivide our analysis to that of rental and return stations, which reflect popular starting and terminal destinations that people elect to choose in Philadelphia.

# Popular Stations - histogram
library(gridExtra)
popular_station <- df %>% group_by(year, month, start_station_id) %>% summarize(rental = n())
colnames(popular_station)[3] <- "id"
popular_station$id <- as.character(popular_station$id)
foo <- df %>% group_by(year, month, end_station_id) %>% summarize(return = n())
colnames(foo)[3] <- "id"
foo$id <- as.character(foo$id)
popular_station <- popular_station %>% full_join(foo, by = c("id" = "id", "year" = "year", "month" = "month"))
rm(foo)
popular_station <- tidyr::gather(popular_station, type, count, c(4,5))
colnames(popular_station)[3] <- "id"
popular_station[is.na(popular_station)] <- 0
popular_station <- popular_station[popular_station$id!="#N/A",]
station_loc$id <- as.character(station_loc$id)
reduced_station <- popular_station %>% left_join(station_loc, by=c("id"="id")) %>% na.omit
reduced_station$quarter <- (as.numeric(as.character(reduced_station$month)) - 1) %/% 3 + 1
reduced_station <- reduced_station %>% group_by(id, name, lat, long, type, year, quarter) %>% summarize(count = sum(count))
# Earliest quarter: 2015Q2, Latest quarter: 2016Q4
reduced_station$year <- as.numeric(reduced_station$year)
rs_first_rental <- reduced_station[reduced_station$year==2015&reduced_station$quarter==2&reduced_station$type=="rental",]
rs_last_rental <- reduced_station[reduced_station$year==2016&reduced_station$quarter==4&reduced_station$type=="rental",]
rs_first_return <- reduced_station[reduced_station$year==2015&reduced_station$quarter==2&reduced_station$type=="return",]
rs_last_return <- reduced_station[reduced_station$year==2016&reduced_station$quarter==4&reduced_station$type=="return",]
rs_first_rental <- rs_first_rental[order(-rs_first_rental$count),][1:10,]
rs_last_rental <- rs_last_rental[order(-rs_last_rental$count),][1:10,]
rs_first_return <- rs_first_return[order(-rs_first_return$count),][1:10,]
rs_last_return <- rs_last_return[order(-rs_last_return$count),][1:10,]
rs_first_rental$name <- factor(rs_first_rental$name, levels = rs_first_rental$name[order(rs_first_rental$count)])
rs_last_rental$name <- factor(rs_last_rental$name, levels = rs_last_rental$name[order(rs_last_rental$count)])
rs_first_return$name <- factor(rs_first_return$name, levels = rs_first_return$name[order(rs_first_return$count)])
rs_last_return$name <- factor(rs_last_return$name, levels = rs_last_return$name[order(rs_last_return$count)])
p1 <- ggplot(rs_first_rental, aes(x = name, y = count)) +
  geom_bar(stat = 'identity', color="blue", fill="blue") +
  geom_text(aes(label=count), hjust=-0.05, size=2.5) +
  coord_flip() +
  ggtitle("Most Popular Rental Station 2015Q2") +
  xlab("Frequency") +
  ylab("Station") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))
p2 <- ggplot(rs_last_rental, aes(x = name, y = count)) +
  geom_bar(stat = 'identity', color="blue", fill="blue") +
  geom_text(aes(label=count), hjust=-0.05, size=2.5) +
  coord_flip() +
  ggtitle("Most Popular Rental Station 2016Q4") +
  xlab("Frequency") +
  ylab("Station") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))
p3 <- ggplot(rs_first_return, aes(x = name, y = count)) +
  geom_bar(stat = 'identity', color="blue", fill="blue") +
  geom_text(aes(label=count), hjust=-0.05, size=2.5) +
  coord_flip() +
  ggtitle("Most Popular Return Station 2015Q2") +
  xlab("Frequency") +
  ylab("Station") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))
p4 <- ggplot(rs_last_return, aes(x = name, y = count)) +
  geom_bar(stat = 'identity', color="blue", fill="blue") +
  geom_text(aes(label=count), hjust=-0.05, size=2.5) +
  coord_flip() +
  ggtitle("Most Popular Return Station 2016Q4") +
  xlab("Frequency") +
  ylab("Station") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))
p1 <- ggplot_gtable(ggplot_build(p1))
p2 <- ggplot_gtable(ggplot_build(p2))
p3 <- ggplot_gtable(ggplot_build(p3))
p4 <- ggplot_gtable(ggplot_build(p4))
maxWidth = grid::unit.pmax(p1$widths[2:3], p2$widths[2:3], p3$widths[2:3], p4$widths[2:3])
p1$widths[2:3] <- maxWidth
p2$widths[2:3] <- maxWidth
p3$widths[2:3] <- maxWidth
p4$widths[2:3] <- maxWidth
# Plot histograms for selected quarters
grid.arrange(p1, p3, p2, p4, heights = c(3, 3))

In the above graph, we see that the most popular stations are somewhat invariant to time. We chose the first and latest quarters to determine most popular rental and return stations. In all cases, the most popular rental and return station was Rittenhouse Square. This is intuitive as it is not only a popular tourist destination, being hailed as “one of the finest urban public spaces in the United States,” (at least according to Wikipedia) but it is also in close proximity to the epicenter of Philadelphia. More specifically, in Q2 of 2015, the top five most popular rental and return stations were all the same, with the exact same ranking. Thus, it would seem that there is no preference for dropoff or pickup stations, but simply hubs for bike users. In Q4 of 2016, we observe many of the same return and and rental stations, but differences exist. For instance, Philadelphia Museum of Art (PMOA) is no longer in the top ten most popular return or rental stations whereas University City station is, which was not in Q2 of 2015. These variability are difficult to explain based on the dataset alone, but we surmise a couple of possible causes. First, the demographic of Indego users has shifted over time (from intrigued users to a those using it for routine trips). Second, seasonality may affect popular rental/return stations. There is a larger tourist cohort during the summer so that may be why PMOA is so high up in the rankings for Q2 of 2015. These are all conjectural, and it may serve as an interesting avenue for further exploration. However this is beyond the scope of our project.

We postulate that the popularity of certain stations across time is due to the fact they are in areas of heavy pedestrian traffic (e.g. city centers), and they are in close proximity to one another but outside comfortable walking range. We thus put this idea to the test by considering spatial patterns and distances amongst these more popular stations.

# Popular stations - maps
rs_first_return$type <- "Return"
rs_last_return$type <- "Return"
rs_first_rental$type <- "Rental"
rs_last_rental$type <- "Rental"
Q2Y15 <- rbind(rs_first_return, rs_first_rental)
Q4Y16 <- rbind(rs_last_return, rs_last_rental)
pal <- colorFactor(c("navy", "red"), domain = c("Return", "Rental"))
# Purple: popular for both return and rental
# Red: only for rental
# Blue: only for return
# Size: total frequency within the quarter
# for Q2 of 2015
leaflet(Q2Y15) %>%
  addTiles() %>%
  addProviderTiles('CartoDB.Positron') %>%
  setView(lng=-75.163763, lat=39.952568, zoom = 13) %>%
  addCircleMarkers(
    radius = ~sqrt(count)/10,
    color = ~pal(type),
    stroke = FALSE, fillOpacity = 0.5, 
    popup = ~name
  )

# for Q4 of 2016
leaflet(Q4Y16) %>%
  addTiles() %>%
  addProviderTiles('CartoDB.Positron') %>%
  setView(lng=-75.163763, lat=39.952568, zoom = 13) %>%
  addCircleMarkers(
    radius = ~sqrt(count)/10,
    color = ~pal(type),
    stroke = FALSE, fillOpacity = 0.5, 
    popup = ~name
  )


In the above two interactive maps, the color purple represents popular rental and return stations, red for rental, and blue for return. The size of the dot denotes the total frequency of rides within the quarter (top: Q2 of 2015, bottom: bottom: Q4 of 2016). Crudely in line with our hypothesis, most of the most popular stations (by dot size) are concentrated in the epicenter of Philadelphia, which is roughly between Rittenhouse Square and Chinatown. We also see that PMOA is off in the distance to the northeast, but was a popular rental and return station in Q2 of 2015. This also corresponds to our theory that summer made this a popular hub for bike users. In Q4 of 2016, the shift in clustering of popular stations revolves around the two shores of Schuylkill River. This may corroborate our observation that people use it as a means to cross the river for routine commutes (e.g. between University City/Powelton Village and Rittenhouse Square) that are out of walking range but close enough for a bike ride. This is purely speculative but makes for an interesting idea to further investigate. We will touch on this a bit more in the next section.

Daily traffic

We put the theory of using Indego as a form of commute to the test, using daily traffic as a proxy for measurements. We picked Q2 of 2016, which is the quarter with the second most bike rides. We ideally would have liked to chosen Q3 of 2016 (most recent summer quarter), which is the most popular quarter, but this time period overlaps with summer vacation for many students. Since we did not want to ignore this particular demographic when analyzing the use of bike trips as a form of commute, we thus settled on Q2 of 2016. We considered average hourly uses of Indego bikes per day for every day of Q2 of 2016. We then plotted this on a 24-hour time-scale, which is shown below. As illustrated by the plot, there are drastic differences between weekday and weekend usage across any standard 24-hour window. On weekdays, peak periods of bike trips were around 8:00 AM and 5:00 PM, which correspond to the time frames that most individuals commute respectively to and from work and school. On weekends, however, peak usage occured during the afternoon. Moreover, more bike trips were observed on weekdays for almost any given time interval than that of weekends. Thus, this may potentially demonstrate that Indego has become a means for everyday commute in Philadelphia.

# 2. Peak Periods (group by hour, weekdays/weekends, season)
# 2.1 Weekdays/weekends histogram of start (rental) hours for all stations in 2016Q2
library(lubridate)
Q2Y16 <- df[df$year==2016 & df$month %in% c("04","05","06"), ]
Q2Y16$day_type <- chron::is.weekend(Q2Y16$date)
Q2Y16$day_type <- ifelse(Q2Y16$day_type == 1, "Weekends", "Weekdays")
n_wknds <- length(unique(Q2Y16[Q2Y16$day_type=="Weekends",14]))
n_wkdys <- length(unique(Q2Y16[Q2Y16$day_type=="Weekdays",14]))
wknds_peak <- Q2Y16[Q2Y16$day_type=="Weekends",] %>% group_by(s_hour,day_type) %>% summarize(rental_freq = n()/n_wknds)
wkdys_peak <- Q2Y16[Q2Y16$day_type=="Weekdays",] %>% group_by(s_hour,day_type) %>% summarize(rental_freq = n()/n_wkdys)
peak <- rbind(wknds_peak, wkdys_peak)
rm(wknds_peak, wkdys_peak)
peak$s_hour <- as.numeric(as.character(peak$s_hour))
foo1 <- c(0,4,8,12,16,20)
foo2 <- c("12am","4am","8am","12pm","4pm","8pm")
ggplot(peak, aes(s_hour, rental_freq)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Peak Rental Hours 2016Q2",
       subtitle="Average Daily Rental Frequency by Hour") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)

We now extend our analysis to all quarterly data available, which is shown in the plot immediately below. Much of the same trends from Q2 of 2016 can be generalized to other quarters. Apart from a few data points, the number of bike trips on weekdays are almost always larger than that of weekends. However, we also observe the aforementioned seasonality of bike trips, in which more bike trips are observed in the summer than in the winter months. This is especially true for trips on weekend (e.g. usage on weekends in Sept of 2015 is by far the largest). Therefore, use of Indego as leisure during summer weekends may be the causal variable of the peaks we observe.

# 2.2 Month aggregation of results
quarter_aggr <- df
#quarter_aggr$quarter <- (as.numeric(as.character(quarter_aggr$month)) - 1) %/% 3 + 1
quarter_aggr$day_type <- chron::is.weekend(quarter_aggr$date)
quarter_aggr$day_type <- ifelse(quarter_aggr$day_type == 1, "Weekends", "Weekdays")
quarter_aggr <- quarter_aggr %>% select(year, month, date, day_type)
quarter_aggr <- quarter_aggr %>% group_by(date, year, month, day_type) %>% summarise(count=n())
quarter_aggr <- quarter_aggr %>% group_by(year, month, day_type) %>% summarise(average=mean(count))
quarter_aggr$no <- (0:(nrow(quarter_aggr)-1)) %/% 2 + 1
foo1 <- seq(1,21,by=3)
foo2 <- c("Apr 2015","Jul 2015","Oct 2015", "Jan 2016", "Apr 2016","Jul 2016","Oct 2016")
ggplot(quarter_aggr, aes(no, average)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Monthly Rental Pattern",
       subtitle="Daily Average of Rental Frequencies") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2) +
  theme(axis.text.x = element_text(angle = 30, hjust = 1))

It is important to note that one of the more glaring deficiencies of our raw dataset is the lack of historical data; we only have data from April 2015 onwards. It thus makes it more difficult to draw conclusive remakrs regarding yearly or even seasonal patterns that require data over longer time periods. Nonetheless, several important trends can still be observed over the time period for which the dataset is available.

For instance, let us take a look at monthly bike trips as determined from the number of average trips per day, shown below. Such an aggregation, as opposed to taking the monthly average as is, was done to smooth out daily fluctuations that can be considered noise as we make our time scale more granular. We observe two main themes. First, there seems to be particular months in which bike trips are more and less frequent. Across any given calendar year, there are more bike rides through the summer, ultimately peaking in September. Bike trips decline thereafter, hitting low points in December/January. This is intuitive as good weather is correlated with how willing and practical bike rides function as a means of transportation. Thus, with abysmal weather conditions in the winter, people are probably less willing to use Indego bikes. Second, we observe Indego’s popularity increase through time based on the number of bike trips in a given month as compared to the same month in a preceding year. In Sept 2016, usage was at an all-time high, with more than 2600 rides per day (cf. approximately 2250 rides in Sept 2016). In Dec 2016, bike rides were roughly a little higher than 1000 rides per day (cf. 600 in Dec 2015). It should be noted that confounding factors such as weather conditions in a given year (e.g. El Nino) and the insufficient historical data we have on-hand may somewhat render our results questionable. We, however, believe this is not the case since more ecofriendly means of transportation should theoretically become more popular due to growing environmental sentiments in the mass population.

# Monthly Bike Trips
df_count_by_month <- df %>% group_by(year, month) %>% summarize(count = n())
df_average_by_month <- df %>% group_by(year, month, day) %>% summarize(count = n())
df_average_by_month <- df_average_by_month %>% group_by(year, month) %>% summarize(avg = mean(count))
df_count_by_month$date <- zoo::as.yearmon(paste(df_count_by_month$year, df_count_by_month$month), "%Y %m")
df_count_by_month$average <- df_average_by_month$avg
df_count_by_month$no <- 1:nrow(df_count_by_month)
df_count_by_month <- df_count_by_month[,-c(1,2)]
rm(df_average_by_month)
df_count_by_month <- tidyr::gather(df_count_by_month, type, value, c(1,3))
foo1 <- c(0, 500, 1000, 1500, 2000, 2500, 3000)
foo2 <- c(3, 6, 9, 12, 15, 18, 21)
ggplot() +
  geom_line(aes(no, value, color="blue"),
            df_count_by_month[df_count_by_month$type=="average",]) +
  geom_point(size = 1.1) +
  ggtitle("Monthly Indego Bike Trips\n(Average per day)") +
  xlab("Month") +
  ylab("Count") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(axis.text.x = element_text(angle = 30, hjust = 1),legend.position="none") +
  scale_y_continuous(limits = c(0, 3000), breaks = foo1) +
  scale_x_continuous(breaks = foo2, labels = df_count_by_month$date[foo2]) +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))

As an interesting tangent, we examine the most popular routes based on a daily average of trips betweeen all stations across available quarters. We see that the most popular routes frequented by most users are those between hubs (i.e. already popular stations). In the interactive map below, we plot the Euclidean distance of the top five most popular routes. For instance, one such route is that between University Station and 23rd and South (i.e. downtown). It is worth noting that most of these routes are either across the downtown area or across the Schuylkill River, which somewhat substantiates the assertion that popular routes are a means for commute.

######## Popular Routes ######## 
# Content
# Most Popular 20 Routes
# Single dots means rent from these stations and return to these stations as well
popular_route <- df %>% group_by(date, year, month, start_station_id, end_station_id) %>% summarise(count=n())
popular_route <- popular_route %>% group_by(start_station_id, end_station_id) %>% summarise(avg=mean(count)) %>% na.omit
popular_route <- popular_route[order(-popular_route$avg),][1:20,]
# Extract all points
loc <- c(popular_route$start_station_id, popular_route$end_station_id) %>% unique %>% as.integer
station_loc$id <- as.numeric(station_loc$id)
dot_loc <- data.frame(id=loc) %>% left_join(station_loc, by=c("id"="id")) %>% na.omit
l1 <- leaflet(dot_loc) %>%
  addTiles() %>%
  addProviderTiles('CartoDB.Positron') %>%
  setView(lng=-75.163763, lat=39.952568, zoom = 13) %>%
  addCircleMarkers(
    color = "navy",
    stroke = FALSE, fillOpacity = 0.5, 
    popup = ~name
  )
popu <- popular_route[popular_route$start_station_id!=popular_route$end_station_id,]
dot_loc2 <- dot_loc
colnames(popu)[1] <- "id"
popu$id <- as.numeric(popu$id)
popu <- popu %>% left_join(dot_loc2, by=c("id"="id")) %>% na.omit
colnames(popu) <- c("sid", "id", "avg", "s_lat", "s_long", "name", "date")
dot_loc2$id <- as.numeric(dot_loc2$id)
popu <- popu %>% left_join(dot_loc2, by=c("id"="id")) %>% na.omit
colnames(popu) <- c("sid", "id", "avg", "s_lat", "s_long", "s_name", "date", "e_lat", "e_long", "e_name", "date_e")
for(i in 1:nrow(popu)){
  l1 <- addPolylines(l1, lat = as.numeric(popu[i, c(4, 8)]), lng = as.numeric(popu[i, c(5, 9)]))
}
l1

Based on the above findings, we can say with a certain level of confidence that the volume of bike trips is an important indicator of Indego’s utility rate over different cross-sections of time. However, it does little to tell us how (not how much) Indego is being utilized. With this in mind, we motivate our next portion of analysis.

Since the inherent purpose of bike share networks are to mediate relatively short commutes, we want to examine the length of each trip. We surmise that shorter trips (0-30 minutes) should be the ideal range since anything beyond that is simply exercise and other transportation methods become increasingly more appealing. We consequently explore features related to the length of bike rides next.

Trip durations

In this section, we try to elucidate the more salient patterns pertaining to trip durations. From the two histograms below, which illustrate the durations of all trips in Q2 and Q4 of 2016 across weekdays and weekends, we observe that most trips taken were under 15 minutes. These two particular quarters were chosen since they are the latest quarters that also offers a contrast in seasonality. During weekdays for both quarters, shorter commutes (< 15 minutes) were the overwhelming majority of trip durations. They make up more than 70% of the total count. The same is true for weekends, but to a lesser extent. This may once again be an indicator that bikes used during weekdays are for commute (e.g. to and from work and school) so they are inherently shorter, whereas bikes are used for more recreational purposes on weekends so their time of use is naturally longer. The latter is corroborated by the fact that during summer weekends, the frequency of longer trips outweighed that of shorter trips.

If we ignore any differentiation of trip durations between weekdays and weekends, we observe much of the same trends. For Q4 of 2016, the latest quarterly data available to us, the most prevalent duration of bike rides lie between 5 and 10 minutes. In fact, the overwhelming majority of trips are less than 30 minutes. On the contrary, we see that long trips are actually outliers in our dataset. Trips between two and four hours only make up 0.9% of all rides in Q4 of 2016, and trips exceeding 4 hours make up 0.7% of the total.

# 1. General Pattern Visualization
# 1.1 Trip Duration Distributions (dimension of time)
# Weekdays/weekends in 2016Q2 and 2016Q4
library(ggthemes)
Q2Y16 <- df[df$year==2016 & df$month %in% c("04","05","06"), ]
Q2Y16$day_type <- chron::is.weekend(Q2Y16$date)
Q2Y16$day_type <- ifelse(Q2Y16$day_type == 1, "Weekends", "Weekdays")
Q4Y16 <- df[df$year==2016 & df$month %in% c("10","11","12"), ]
Q4Y16$day_type <- chron::is.weekend(Q4Y16$date)
Q4Y16$day_type <- ifelse(Q4Y16$day_type == 1, "Weekends", "Weekdays")
Q2Y16$duration_cut <- cut(as.numeric(Q2Y16$duration), c(0,5,10,15,20,30,60,120,240,1440))
Q4Y16$duration_cut <- cut(as.numeric(Q4Y16$duration), c(0,5,10,15,20,30,60,120,240,1440))
Q2Y16_dur <- Q2Y16 %>% group_by(day_type, duration_cut, date) %>% summarise(count=n())
Q2Y16_dur<- Q2Y16_dur %>% group_by(day_type, duration_cut) %>% summarise(average=mean(count))
Q2Y16_dur$Quarter <- "2016Q2"
Q4Y16_dur <- Q4Y16 %>% group_by(day_type, duration_cut, date) %>% summarise(count=n())
Q4Y16_dur<- Q4Y16_dur %>% group_by(day_type, duration_cut) %>% summarise(average=mean(count))
Q4Y16_dur$Quarter <- "2016Q4"
dur_cut <- rbind(Q2Y16_dur, Q4Y16_dur)
rm(Q2Y16_dur, Q4Y16_dur, Q2Y16, Q4Y16)
foo <- c("<5min", "5-10min", "10-15min", "15-20min", "20-30min", "30-60min", "1-2hr", "2-4hr", ">4hr")
ggplot(dur_cut, aes(duration_cut, average)) +
  geom_bar(aes(fill=Quarter), stat = 'identity', position = position_dodge()) +
  facet_grid(. ~ day_type, scales = "free") +
  labs(title="Daily Trip Duration Histogram",
       subtitle="For 2016Q2 and 2016Q4",
       xlab="") +
  scale_color_fivethirtyeight(" ") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_discrete(labels = foo) +
  theme(axis.text.x = element_text(angle = 30, hjust = 1)) + 
  scale_fill_manual(values = c("#0072B2", "#CC79A7"))

We next examine the median rental duration for each station, particularly for Q2 to Q4 of 2016. We chose this range since it was the only range of contiuous quarterly data that had complete station entries for trip durations. In the interactive map below, larger circles correspond to larger values with respect to trip durations. The color scheme remains the same as previously mentioned. Therefore, a larger red circle implies that a bike user rented from a particular station and rode their bike for a longer duration. We see that larger red circles correspond to stations in the outskirts of Philadelphia. Therefore, they either used the bike for a longer commute or they used for leisurely purposes. For instance, one of the larger red circles denotes PMOA, which we previously mentioned as a popular station during the summer and weekends. However, it is impossible to differentiate between the two based on our dataset alone. It is however intriguing in nature.

# 1.2 Median Rental Duration for Each Station (2016 Q2-Q4) (dimension of space)
# choose 2016 Q2-Q4 since all stations have complete data in this period
station_ana2 <- df[df$year==2016 & df$month %in% c("04","05","06","07","08","09","10","11","12"), ]
median_stat <- station_ana2 %>% group_by(start_station_id) %>% summarise(med=median(duration))
# Plot in a map, dot size = median rental time
colnames(median_stat)[1] <- "id"
station_loc$id <- as.numeric(station_loc$id)
median_stat$id <- as.numeric(median_stat$id)
median_stat <- median_stat %>% left_join(station_loc, by=c("id"="id")) %>% na.omit
content <- paste("<b>", as.character(median_stat$name), "</b><br>",
                 "Median Rental Time: ", median_stat$med, "minutes", "<br>")
median_stat$chosen <- median_stat$id %in% c("3020", "3050", "3124")
median_stat$chosen <- ifelse(median_stat$chosen == 1, "Chosen", "Unchosen")
pal <- colorFactor(c("navy", "red"), domain = c("Chosen", "Unchosen"))
leaflet(median_stat) %>%
  addTiles() %>%
  addProviderTiles('CartoDB.Positron') %>%
  setView(lng=-75.163763, lat=39.952568, zoom = 13) %>%
  addCircleMarkers(
    radius = ~med/2,
    color = ~pal(chosen),
    stroke = FALSE, fillOpacity = 0.5, 
    popup = ~content
  )


The dots colored in navy are those that we chose to further analyze, which we present here. They correspond to the following stations:

  • 3020: University City Station
  • 3050: 9th & Arch (downtown)
  • 3124: Race Street Pier (away from university and downtown)
# 2. Station-level rental time
# Select stations
# Three locations that could have totally different travel patterns:
# 3020: University City Station, global avg duration 17.8
# 3050: 9th & Arch, (downtown), global avg duration 22.4
# 3124: Race Street Pier, (away from university and downtown), global avg duration 43.4
# 2.1 Over hour of a day
station_ana <- df[df$year==2016 & df$month %in% c("04","05","06","07","08","09","10","11","12"), ]
foo <- station_ana %>% group_by(start_station_id) %>% summarise(average=mean(duration)) # select stattions
station_ana <- station_ana[station_ana$duration<240,] # !! Truncate duration at 4 hrs to remove outliers
station_ana <- station_ana[station_ana$start_station_id %in% c("3020", "3050", "3124"),]
station_ana$day_type <- chron::is.weekend(station_ana$date)
station_ana$day_type <- ifelse(station_ana$day_type == 1, "Weekends", "Weekdays")
foo <- station_ana %>% group_by(start_station_id) %>% summarise(count=n()) # check total trips
station_ana <- station_ana %>% group_by(start_station_id, day_type, s_hour) %>% summarise(avg_dur=mean(duration))
foo <- data.frame(start_station_id = rep(c("3020", "3050", "3124"),each=48),
                  day_type = rep(rep(c("Weekdays", "Weekends"),each=24),3),
                  s_hour = rep(0:23,6))
station_ana$s_hour <- as.numeric(as.character(station_ana$s_hour))
station_ana$start_station_id <- as.character(station_ana$start_station_id)
station_ana$day_type <- as.character(station_ana$day_type)
station_ana <- station_ana %>% right_join(foo, by=c("start_station_id"="start_station_id",
                                                    "day_type"="day_type",
                                                    "s_hour"="s_hour"))
station_ana[is.na(station_ana)] <- 0
foo1 <- c(0,4,8,12,16,20)
foo2 <- c("12am","4am","8am","12pm","4pm","8pm")
ggplot(station_ana[station_ana$start_station_id=="3020",], aes(s_hour, avg_dur)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Average Trip Duration 2016 Q2-Q4",
       subtitle="At (3020) University City Station, Total Trips: 11920") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)

ggplot(station_ana[station_ana$start_station_id=="3050",], aes(s_hour, avg_dur)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Average Trip Duration 2016 Q2-Q4",
       subtitle="At (3050) 9th & Arch, Total Trips: 6121") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)

ggplot(station_ana[station_ana$start_station_id=="3124",], aes(s_hour, avg_dur)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Average Trip Duration 2016 Q2-Q4",
       subtitle="At (3124) Race Street Pier, Total Trips: 3729") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)

In the first plot (University City Station), we consdier the station close to a university. We see that on average, longer trips occur on the weekend. In fact, none of the trips on weekdays exceed 25 minutes. We also observe that in both cases, longer trips tend to occur during the day, whereas shorter trips occur in the early morning and towards night time. In the second plot delineating the downtown station (9th & Arch), we see much of the same trend as that of the previous case. However, the discrepancy between trip durations across a 24-hour window on the weekends and the weekdays is much more pronounced. In our last plot above (Race Street Pier), we consider a station far from a university or downtown, which we can consider to be “more rural.” Here, trips on weekdays and weekends mirror one another, showing they are closely related. Moreover, trips from this more rural station are all much longer, most exceeding 20 minutes irrespective of time intervals during the week. Therefore, it would seem that different stations based on their geographical location serve a different purpose for bike riders. Stations in more densely populated areas tend to favor shorter rides, whereas longer rides are more prevalent in more remote locations. It is important to note that the magnitude of number of trips may introduce some sort of counfounder on our insights (e.g. data imbalance), but we ignore this here, since it is difficult to address this issue given the limited historical data we have. This is a problem that would be solved naturally as more data becomes available.

# 2.2 Over month of a year
station_ana2 <- df[df$year==2016 & df$month %in% c("04","05","06","07","08","09","10","11","12"), ]
station_ana2 <- station_ana2[station_ana2$start_station_id %in% c("3020", "3050", "3124"),]
station_ana2 <- station_ana2[station_ana2$duration<240,] # !! Truncate duration at 4 hrs to remove outliers
station_ana2 <- station_ana2 %>% group_by(start_station_id, month) %>% summarise(med=median(duration))
colnames(station_ana2)[1] <- "id"
station_ana2$id <- as.numeric(station_ana2$id)
station_ana2 <- station_ana2 %>% left_join(station_loc, by=c("id"="id"))
station_ana2$month <- as.numeric(as.character(station_ana2$month))
foo1 <- 4:12
foo2 <- c("Apr","May","Jun","Jul","Aug","Sept","Oct","Nov","Dec")
ggplot(station_ana2, aes(month, med)) +
  geom_line(aes(color=name)) +
  geom_point(size = 1.1, aes(color=name)) +
  labs(title="Median of Trip Duration",
       subtitle="Over Months (2016 Q2-Q4)") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)

We realize it may be difficult to examine the trends of all three plots concurrently so we thus plot the median trip duration of all three stations in one single plot, which is shown above. It is interesting to note that over time, shorter rides become the prevalent theme. This is true for all three stations. However, the more rural station is still characterized by longer trips relative to the other two stations that are close to a university and city center.

Thus far, we have examined how and how much Indigo is utilized. However, we still do not know how users select to utilize Indego. For instance, which methods of payment are more common? Is there a change in preference over time? We seek to answer these questions in the next and final section of our analysis.

User profiles

In this section, we examine the attributes related to Indego bike users. Since the dataset provides only two dimensions regarding this aspect, we use the conventional mosaic plot to examine the proportions of different combinations of Indego users. These two features are passholder types (Indego30, IndegoFlex, and walkup) and type of trips (one-way, roundtrip).

Here is some more relevant information regarding the different passholder types:

  • Indego30: $15/month, unlimited one hour trips
  • IndegoFlex: $10/year base fee, and $4/hour
  • Walk-up: $8/hour
# 1.1 Mosaic plot of overall data
all_data_mosaic <- df %>% group_by(passholder_type, trip_route_category) %>% summarize(count = n())
colnames(all_data_mosaic) <- c("Pass", "Trip", "count")
library(vcd)
vcd::mosaic(xtabs(count ~ Trip + Pass, data = all_data_mosaic),
            direction = c("v", "h"),
            gp = gpar(fill = c("light green", "light blue", "light yellow")),
            spacing = spacing_highlighting,
            labeling = labeling_border(rot_labels = c(0,90,0,0)))

For all available quarters, we can see that the majority of users drastically favor one-way over round trips, and Indego30 passes over walkup payment plans. IndegoFlex is the least popular option regardless of route type. In fact, the combination of one-way trips with Indego30 is overwhelming characteristic, accounting for >50% of all users, followed by one-way with walk-ups. However, this is a extremely rough representation of user profiles since we ignore how user profiles vary over time. We examine this next.

Let us examine the most popular option (Indego30 x one-way), as shown below. We see that this option has become more popular at a steady rate, in which more than 75% of users choose this option in the lastest quarter.

# 1.2 Type of passholders, frequency of round-trips (for month)
# for most popular type of trip route and pass type
# 2. Tendency of Moving to Most Popular Category
# Plot of the percentage of (passholder_type, trip_route_category) over quarters
# For most popular category
perc_category <- df %>% group_by(passholder_type, trip_route_category, year, month) %>% summarize(count = n())
perc_category$quarter <- (as.numeric(as.character(perc_category$month)) - 1) %/% 3 + 1
perc_category <- perc_category %>% group_by(year, quarter, passholder_type, trip_route_category) %>% summarise(count_qt=sum(count)) 
perc_category$type <- paste(perc_category$trip_route_category, "x", perc_category$passholder_type)
perc_category <- perc_category %>% group_by(year, quarter) %>% mutate(total=sum(count_qt))
perc_category$perc <- round(100*(perc_category$count_qt / perc_category$total),2)
perc_category <- perc_category[,-c(3:5,7)]
perc_category$no <- rep(1:7,each=6)
foo1 <- 1:7
foo2 <- c("2015Q2","2015Q3","2015Q4", "2016Q1", "2016Q2","2016Q3","2016Q4")
foo3 <- c("One Way x Indego30")
tmp1 <- perc_category[perc_category$type %in% foo3,]
ggplot(tmp1, aes(no, perc)) +
  geom_line(aes(colour=type)) +
  geom_point(aes(colour=type),size = 1.1) +
  labs(title="Percentage Change Over Quarters",
       subtitle="For Most Popular Category\nOne Way x Indego30") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2) +
  scale_y_continuous(limits = c(0, 100))

We see from the previous mosaic plot that round trips are not popular on average. We want to test whether there does indeed exist a decline in popularity over time. We see from the percentage change over quarters shown below that regardless of pass types, round trips make up an increasingly smaller fraction of the user demographic. We also postulate that the already unpopular combination of round trips and Indego Flex would make up an increasingly smaller proportion as well since they are more frequent travellers that would gradually move towards cheaper plans. This however is speculative and is contingent on a robust, large enough longitudinal dataset, which we do not have. Nontheless, this problem, albeit interesting, is beyond the scope of our analysis.

# For roundtrips
foo4 <- c("Round Trip x IndegoFlex", "Round Trip x Indego30", "Round Trip x Walk-up")
tmp2 <- perc_category[perc_category$type %in% foo4,]
ggplot(tmp2, aes(no, perc)) +
  geom_line(aes(colour=type)) +
  geom_point(aes(colour=type),size = 1.1) +
  labs(title="Percentage Change Over Quarters",
       subtitle="Over Categories") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2) +
  scale_y_continuous(limits = c(0, 7.15))

Based on the above analysis on user profiles, we see that as bike sharing becomes more popular over time in Philadelphia, more users opt for more cost-efficient payment plans, which is by far Indego30. The rise of one-way trips also conjecturally demonstrate that users are using Indego more frequently as a means of transportation over short distances or time periods. Obviously, we ignore many important factors related to user profiles. For instance, is there a preference for those renting from particular stations or during particular months? This is a natural extension of our analysis that we would like to further explore time permitting. However, we believe that our current set of analysis on user profiles presented here serves to provide an overarching representation of user profiles.

Conclusion

In this project, we examined data related to Philadelphia’s bike share program, Indego. We examined all available quarterly data (Q2 of 2015 to Q4 of 2016) to identify noteworthy patterns and trends related to the various ways that Indego is utilized, mainly pertaining to station usage, daily traffic, trip durations, and user profiles. We found that popular rental and return stations are concentrated around locations that are densely populated (e.g. in close proximity to city centers or university campuses). We also note that seasonality affects the usage rate of certain Indego stations. Using daily traffic as a proxy to determine whether or not Indego is used as a means of commute, we also found an increase in frequency of bike trips on weekdays that conveniently overlaps with time periods when most people commute to and from work or school. Moreover, we observe a higher volume of bike trips during the weekday, all of which conjecturally demostrate the vercaity of our aforementioned claim.

We yet again observe trends of seasonality in daily traffic, in which volume during the summer far exceeds that of winter months. We also examined trip durations, and found that shorter trips (0-15 minutes) are the norm, but this is less evident for weekends than weekdays. Finally, we evaluated the proportions of Indego’s user profiles and found that most users prefer monthly payments and one-way trips. We also found that over time, more individuals are opting for yearly memeberships, which may speak to the fact that Indego is becoming a more popular means of short-range commute in Philadelphia.

It is worth mentioning that our analyses is extremely broad in nature. We would have liked to further explore certain trends and interaction effects. For instance, how does certain user profiles and stations correlate with each other? Are there any neighborhoods that have extremely high volumes of traffic? How does this vary monthly, quarterly, or even yearly? We would have liked to compared results across similar cities, and how they compare and constrast (e.g. with NYC’s citibike). Time permitting, these are the more imminent avenues we would have liked to explore.

Another caveat that should be brought to light is that the lack of historical data also makes it difficult to draw robust conclusions on year-to-year variability or even corroborate the significance of seasonal fluctuations. This problem obviously becomes less relevant as more quarterly data becomes available, but is a major obstacle in our current analysis for long-term time frames.

Nonetheless, with the data available, we believe that the exploratory data analysis and visualizations that we provide here serve as a good first-order approximation of salient trends related to Indego and the ways it is being effectively utilized. We hope this serves as a pragmatic catalyst for other readers interested in the Indego dataset, or using the insights and techinques provided here to explore other similar datasets.

Appendix: Shiny app

We also created a Shiny application for real-time status updates for all active Indego stations. This is purely supplemental, and we provide snapshots here, and instructions to run our application in our Github repo linked above. You can follow the instructions on the website to run it on a local machine.

---
title: "STAT 5702: Final Project"
output: html_notebook
---

## Introduction

#### Motivation
 
Indego is the de facto bike sharing network in Philadelphia. It has become a popular, more eco-friendly alternative for short trips across the city since launching in late April of 2015. However, little has been done to analyze the publicly available data associated with Indego. Thus, we think it would be beneficial to the public if we quantify and assess several pertinent features and patterns related to Indego, generating insights on how the bike sharing network is utilized across the city. More specifically, we want to more closely study and answer questions related to the following four themes:

* Station usage (usage rate, peak periods, popular rental/return stations, etc.) 
* Daily traffic (popular routes, volume across different sections, travel patterns, etc.)
* Trip durations (length of trips, median rental/return time, etc.)
* User profiles (type of passholders, frequency of round-trips, seasonality, etc.)

Each of the above four features will be explained in finer details in subsequent sections. To realize this objective, we have created static plots (shown in this notebook) to describe salient themes as well as a Shiny app ([Github](https://github.com/masonsun/gr5702-proj)), which also contains real-time status updates of all Indego stations across Philadelphia. We will also provide instructions on runnning our app further down in our report.

Our objective starting out was to examine bike share programs in the most densely populated cities across the eastern United States, mainly via exploratory data analysis and visualization techinques. We initially considered NYC's Citibike as a promising candidate, but it has already been extensively analyzed (see [Todd Schneider's blogpost](http://toddwschneider.com/posts/a-tale-of-twenty-two-million-citi-bikes-analyzing-the-nyc-bike-share-system/) for an excellent example). The marginal returns of analyzing such a dataset are thus minimal, if not, nil. As a result, we decided on Philadelphia, primarliy for two reasons: it is the second largest city satisfying our predefined criteria (after NYC), and also has a relatively new bike share program. Thus, we thought the Indego dataset would be an ideal choice. It would also be interesting to use our dataset to compare and contrast New York and Philadelphia's bike share programs.

#### Data Source

The dataset is publicly available on [Indego's website](https://www.rideindego.com/about/data/). It can be found by clicking the <i>About</i> panel and then selecting <i>Data</i>. All quarterly data are listed under the section <i>The Data</i>, in which each station's information is available through the <i>Station Table</i> hyperlink. In addition, real-time data used in our Shiny app is available under the section <i>Indego Station Status</i> via the GeoJSON hyperlink.

## Team

1. Kejia Shi (ks3403): data processing, exploratory data analysis, visualization (static and Shiny plots), writing/editing report.
2. Mason Sun (zs2321): data processing, exploratory data analysis, developing front/backend of Shiny app, writing/editing report.
3. Hung-Shi Lin (hl2997): proofread.

## Analysis of Data Quality

#### Quarterly data
For quarterly data, we had seven files in total, each representing data between Q2 of 2015 and Q4 of 2016, inclusively. Each contained the same columns:

* trip_id (unique identifier of a bike trip)
* duration (length of trip in seconds or minutes calculated by check-in and check-out time)
* start_time (clock start time and calendar date)
* end_time (clock end time and calenar date)
* start_station_id (unique identifier of starting station)
* start_lat (latitude of starting station)
* start_lon (longitude of starting station)
* end_station_id (unique identifier of end station)
* end_lat (latitude of end station)
* end_lon (longitude of end station)
* bike_id (unique identifier of the bike used during the trip)
* plan_duration (type of pass by number of days, e.g. thirty-day pass is shown as 30)
* trip_route_category (round trip or one-way)
* passholder_type (walkup, IndegoFlex, or Indego30)

We combined these into one data frame (called <i>df</i>) using the code as follows. The description of our preprocessing methods are embedded as comments in the code chunk below. Please see them for more detailed explanations.

```{r, message=FALSE, warning=FALSE}
library(tidyverse)
### read in quarterly data ###
Y16Q4 <- read.csv("data/Indego_Trips_2016Q4.csv", header=T)
Y16Q3 <- read.csv("data/Indego_Trips_2016Q3.csv", header=T)
Y16Q2 <- read.csv("data/Indego_Trips_2016Q2.csv", header=T)
Y16Q1 <- read.csv("data/Indego_Trips_2016Q1.csv", header=T)
Y15Q4 <- read.csv("data/Indego_Trips_2015Q4.csv", header=T)
Y15Q3 <- read.csv("data/Indego_Trips_2015Q3.csv", header=T)
Y15Q2 <- read.csv("data/Indego_Trips_2015Q2.csv", header=T)
indego_stations <- read.csv("data/indego_stations.csv", header=T)
### preprocessing ###
# select currently active stations from most recent quarterly dataset
station_loc <- Y16Q4 %>% dplyr::select(start_station_id, start_lat, start_lon) %>% unique
# create a dataframe that includes the relevant columns for all active stations by left joining on the active stations and all stations, ignoring any rows that contain missing values
station_loc <- dplyr::left_join(station_loc, indego_stations, by = c("start_station_id" = "Station.ID")) %>% na.omit %>% .[, -6]
# rename columns of dataframe
colnames(station_loc) <- c("id", "lat", "long", "name", "date")
# remove whitespaces in the date column
station_loc$date = trim(station_loc$date %>% as.character())

# below are two auxiliary fuction to parse the start/end time and date of each trip, which displays them as a single string. We partition them into separate columns that now display year, month, day, hour, minutes, and date (YYYY-MM-DD).

# for quarters of 2015
parseTime_15 <- function(df) {
  df$s_hour <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%H")
  df$s_min <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%M")
  df$e_hour <- format(as.POSIXct(strptime(df$end_time,"%m/%d/%y %H:%M",tz="")) ,format = "%H")
  df$e_min <- format(as.POSIXct(strptime(df$end_time,"%m/%d/%y %H:%M",tz="")) ,format = "%M")
  df$year <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%Y")
  df$month <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%m")
  df$day <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%d")
  df$date <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%y %H:%M",tz="")) ,format = "%Y-%m-%d")
  return(df)
}

# for quarters of 2016, which are defined differently from 2015
parseTime_16 <- function(df) {
  df$s_hour <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%H")
  df$s_min <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%M")
  df$e_hour <- format(as.POSIXct(strptime(df$end_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%H")
  df$e_min <- format(as.POSIXct(strptime(df$end_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%M")
  df$year <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%Y")
  df$month <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%m")
  df$day <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%d")
  df$date <- format(as.POSIXct(strptime(df$start_time,"%m/%d/%Y %H:%M",tz="")) ,format = "%Y-%m-%d")
  return(df)
}

# pass into respective auxiliary functinos
Y15Q2 <- parseTime_15(Y15Q2)
Y15Q3 <- parseTime_15(Y15Q3)
Y15Q4 <- parseTime_15(Y15Q4)
Y16Q1 <- parseTime_16(Y16Q1)
Y16Q2 <- parseTime_16(Y16Q2)
Y16Q3 <- parseTime_16(Y16Q3)
Y16Q4 <- parseTime_16(Y16Q4)

# combine all quarterly data into a single data frame called df
df <- rbind(Y15Q2, Y15Q3, Y15Q4, Y16Q1, Y16Q2, Y16Q3, Y16Q4)

# standardize duration into units of seconds
# also removes columns not needed in subsequent analyses
dataClean <- function(df) {
  df$duration <- df$duration / 60
  df[, -c(1, 3, 4, 6, 7, 9, 10, 12)]
}
df <- dataClean(df)

# remove individual dataframes
rm(Y15Q2, Y15Q3, Y15Q4, Y16Q1, Y16Q2, Y16Q3, Y16Q4)
```

During our initial exploration of the dataset, we realized some of the respective datasets for the quarterly data contained missing values, which would imply that our final dataframe also contains missing values. We thus try to qualitatively visualize these features, show below:

```{r fig.width=5, fig.height=5}
# ignore columns after passholder_type (col_num = 14) as subsequent columns are defined by us.
library(extracat)
visna(df[,c(1:14)])
```

```{r}
missing_table = table(is.na(df))
```

```{r}
cat("Percentage of missing entries:", missing_table[2]/sum(missing_table)*100)
```

We see that the proportion of missing values is miniscule relative to our whole dataset (approximately 0.0172%). Moreover, they are all related to latitude and longitude data as well as the bike ID. If need be, we can infer the latitude and longitude from the station name or station ID, but we actively chose to ignore rows corresponding to these missing values since we cannot explicate why these values are missing. We thus choose to ignore them altogether.

#### Indego stations data

For the data detailing the information of each station (indego_stations.csv), we had to perform a bit of data preprocessing to get it into an ideal format. The data contained the columns as follows:

* Station ID (unique identifier of a Indego station)
* Station Name (name of corresponding Indego station, usually a street address)
* Go live date (the date that the station started operations)
* Status (its current status, either active or inactive)

There were many leading and trailing whitespaces in the <i>Station Name</i> column, so performed the following to remove these unneeded characters:

```{r}
# load stations data
indego_stations <- read.csv("../data/indego_stations.csv", header=T)
# a function that removes leading and trailing white spaces of a given string
trim <- function (x) gsub("^\\s+|\\s+$", "", x)
# remove whitespaces surrounding each entry in the column of Station.Name
indego_stations$Station.Name <- trim(indego_stations$Station.Name)
```

There were no missing values in this table, so we do not provide plots related to this here.

#### GeoJSON

For real-time status updates of Indego stations, we use the semi-structured [GeoJSON dataset](https://www.rideindego.com/stations/json/) provided by Indego. We allow the user to manually refresh our Shiny app, in which we call the GeoJSON file in real-time to get a visualization update (more specific details in subsequent sections). We used the R packpage [geojsonio](https://cran.r-project.org/web/packages/geojsonio/index.html) to unpack and transform this data into a workable format as follows:

```{r}
library(geojsonio)
station_status <- fromJSON("https://gbfs.bcycle.com/bcycle_indego/station_status.json")
```

## Executive Summary


The primary objective of this report is to derive valuable insights of deploying shared bikes for big cities (in our case, Philadelphia's Indego), and how they are utilized by the public mass. More specifically, we want to understand patterns of usage among different stations and users over time. The exploratory methodologies that we implement on the Indego dataset will generally follow conventional principles of analyzing temporal and spatial features of large datasets. 

Upon retrieving the dataset, we immediately raised the following questions:

1. For the total 105 stations deployed gradually within two years, which stations are the most popular ones? Has the popularity changed over time?

2. For specific stations we have an expection on their daily rental patterns. How do we visualize and examine such patterns?

3. With the geographical data of stations and GeoSON file, how do we draw connections among stations and see rental and return patterns, popular routes, and visualize the real-time station status?

We present here a quick run-through of the plots we present in our main analysis, where we will dive into much finer details. The following plots demonstrate the main findings of our exploratory data analysis.

Firstly, we make general line plots with monthly frequency data and further separate the data into week days and weekends. From the plots, we see clearly the seasonal usage variation.
![](/Users/masonsun/Desktop/STAT 5702/project/pic/freq.png)

Secondly, we pick three different stations located at the university area, downtown and suburban Philadelphia, and draw the average hourly trip durations for week days and weekends in a selected time period. Travel patterns among different types of stations thus becomes evident. For example, the weekends' average trip duration in the afternoon is longer than that of the weekday's for users close to university campuses.
![](/Users/masonsun/Desktop/STAT 5702/project/stations.png)

Thirdly, we utilize `leaflet` to create interactive maps in order to represent the spatial information we can derive from the dataset. We effectively represented the average daily rental and return frequencies among stations using different dot sizes (larger dots implies higher frequency). We embed the real-time GeoJSON file into our `Shiny` app to show real-time updates of station status. In order to jointly analyze the spatial and time dimensions, we pick the ten most popular rental and return stations for two recent quarters to determine differences in trends. Besides the plots below, we also made a map for the five most popular routes (see next part).
![](/Users/masonsun/Desktop/STAT 5702/project/pic/map.png)

Moreover, we analyze the daily trip durations and the user profiles, and uncover the the shift of passholder types and trip route (one-way or round-trip) changes. The most incisive observation we got from this section is that the percentage of the most popular passholder type (Indego30) and one-way trips has gotten more popular. In addition, less people are taking round trips.

![](/Users/masonsun/Desktop/STAT 5702/project/pic/type.png)

Lastly, we create a `Shiny` dashboard to let people explore the dataset as their wishes. That is, they can select stations and date ranges of interest, and also modify parameters of different charts (e.g. number of bins for histograms) to generate customized insights. 
![](/Users/masonsun/Desktop/STAT 5702/project/pic/shiny.png)

## Main analysis 

In this main section, we present our findings of different facets pertaining to the Indego dataset. We subdivide our analysis into four interrelated sections: station usage, daily traffic, trip duration, and user profiles, in that particualr order. We will walk the reader through our thought process and findings, and finally present some concluding remarks.

#### Station usage

In this section, we examine the usage rate among the different Indego stations across Philadelphia. We motivate our analysis from a holistic perspective towards more granular analysis. This structural layout also holds true for subsequent sections. With that in mind, let us first patterns related to some of the more popular Indego stations. We subdivide our analysis to that of rental and return stations, which reflect popular starting and terminal destinations that people elect to choose in Philadelphia.

```{r results="hide",fig.width=5, fig.height=4}
# Popular Stations - histogram
library(gridExtra)
popular_station <- df %>% group_by(year, month, start_station_id) %>% summarize(rental = n())
colnames(popular_station)[3] <- "id"
popular_station$id <- as.character(popular_station$id)
foo <- df %>% group_by(year, month, end_station_id) %>% summarize(return = n())
colnames(foo)[3] <- "id"
foo$id <- as.character(foo$id)

popular_station <- popular_station %>% full_join(foo, by = c("id" = "id", "year" = "year", "month" = "month"))
rm(foo)
popular_station <- tidyr::gather(popular_station, type, count, c(4,5))
colnames(popular_station)[3] <- "id"
popular_station[is.na(popular_station)] <- 0
popular_station <- popular_station[popular_station$id!="#N/A",]
station_loc$id <- as.character(station_loc$id)
reduced_station <- popular_station %>% left_join(station_loc, by=c("id"="id")) %>% na.omit
reduced_station$quarter <- (as.numeric(as.character(reduced_station$month)) - 1) %/% 3 + 1
reduced_station <- reduced_station %>% group_by(id, name, lat, long, type, year, quarter) %>% summarize(count = sum(count))

# Earliest quarter: 2015Q2, Latest quarter: 2016Q4
reduced_station$year <- as.numeric(reduced_station$year)
rs_first_rental <- reduced_station[reduced_station$year==2015&reduced_station$quarter==2&reduced_station$type=="rental",]
rs_last_rental <- reduced_station[reduced_station$year==2016&reduced_station$quarter==4&reduced_station$type=="rental",]
rs_first_return <- reduced_station[reduced_station$year==2015&reduced_station$quarter==2&reduced_station$type=="return",]
rs_last_return <- reduced_station[reduced_station$year==2016&reduced_station$quarter==4&reduced_station$type=="return",]

rs_first_rental <- rs_first_rental[order(-rs_first_rental$count),][1:10,]
rs_last_rental <- rs_last_rental[order(-rs_last_rental$count),][1:10,]
rs_first_return <- rs_first_return[order(-rs_first_return$count),][1:10,]
rs_last_return <- rs_last_return[order(-rs_last_return$count),][1:10,]

rs_first_rental$name <- factor(rs_first_rental$name, levels = rs_first_rental$name[order(rs_first_rental$count)])
rs_last_rental$name <- factor(rs_last_rental$name, levels = rs_last_rental$name[order(rs_last_rental$count)])
rs_first_return$name <- factor(rs_first_return$name, levels = rs_first_return$name[order(rs_first_return$count)])
rs_last_return$name <- factor(rs_last_return$name, levels = rs_last_return$name[order(rs_last_return$count)])

p1 <- ggplot(rs_first_rental, aes(x = name, y = count)) +
  geom_bar(stat = 'identity', color="blue", fill="blue") +
  geom_text(aes(label=count), hjust=-0.05, size=2.5) +
  coord_flip() +
  ggtitle("Most Popular Rental Station 2015Q2") +
  xlab("Frequency") +
  ylab("Station") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))

p2 <- ggplot(rs_last_rental, aes(x = name, y = count)) +
  geom_bar(stat = 'identity', color="blue", fill="blue") +
  geom_text(aes(label=count), hjust=-0.05, size=2.5) +
  coord_flip() +
  ggtitle("Most Popular Rental Station 2016Q4") +
  xlab("Frequency") +
  ylab("Station") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))

p3 <- ggplot(rs_first_return, aes(x = name, y = count)) +
  geom_bar(stat = 'identity', color="blue", fill="blue") +
  geom_text(aes(label=count), hjust=-0.05, size=2.5) +
  coord_flip() +
  ggtitle("Most Popular Return Station 2015Q2") +
  xlab("Frequency") +
  ylab("Station") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))

p4 <- ggplot(rs_last_return, aes(x = name, y = count)) +
  geom_bar(stat = 'identity', color="blue", fill="blue") +
  geom_text(aes(label=count), hjust=-0.05, size=2.5) +
  coord_flip() +
  ggtitle("Most Popular Return Station 2016Q4") +
  xlab("Frequency") +
  ylab("Station") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))

p1 <- ggplot_gtable(ggplot_build(p1))
p2 <- ggplot_gtable(ggplot_build(p2))
p3 <- ggplot_gtable(ggplot_build(p3))
p4 <- ggplot_gtable(ggplot_build(p4))

maxWidth = grid::unit.pmax(p1$widths[2:3], p2$widths[2:3], p3$widths[2:3], p4$widths[2:3])
p1$widths[2:3] <- maxWidth
p2$widths[2:3] <- maxWidth
p3$widths[2:3] <- maxWidth
p4$widths[2:3] <- maxWidth
```

```{r fig.width=7.5, fig.height=5}
# Plot histograms for selected quarters
grid.arrange(p1, p3, p2, p4, heights = c(3, 3))
```

In the above graph, we see that the most popular stations are somewhat invariant to time. We chose the first and latest quarters to determine most popular rental and return stations. In all cases, the most popular rental and return station was [Rittenhouse Square](https://en.wikipedia.org/wiki/Rittenhouse_Square). This is intuitive as it is not only a popular tourist destination, being hailed as "one of the finest urban public spaces in the United States," (at least according to Wikipedia) but it is also in close proximity to the epicenter of Philadelphia. More specifically, in Q2 of 2015, the top five most popular rental and return stations were all the same, with the exact same ranking. Thus, it would seem that there is no preference for dropoff or pickup stations, but simply hubs for bike users. In Q4 of 2016, we observe many of the same return and and rental stations, but differences exist. For instance, Philadelphia Museum of Art (PMOA) is no longer in the top ten most popular return or rental stations whereas University City station is, which was not in Q2 of 2015. These variability are difficult to explain based on the dataset alone, but we surmise a couple of possible causes. First, the demographic of Indego users has shifted over time (from intrigued users to a those using it for routine trips). Second, seasonality may affect popular rental/return stations. There is a larger tourist cohort during the summer so that may be why PMOA is so high up in the rankings for Q2 of 2015. These are all conjectural, and it may serve as an interesting avenue for further exploration. However this is beyond the scope of our project.

We postulate that the popularity of certain stations across time is due to the fact they are in areas of heavy pedestrian traffic (e.g. city centers), and they are in close proximity to one another but outside comfortable walking range. We thus put this idea to the test by considering spatial patterns and distances amongst these more popular stations.

```{r, warning=FALSE, message=FALSE}
# Popular stations - maps
rs_first_return$type <- "Return"
rs_last_return$type <- "Return"
rs_first_rental$type <- "Rental"
rs_last_rental$type <- "Rental"

Q2Y15 <- rbind(rs_first_return, rs_first_rental)
Q4Y16 <- rbind(rs_last_return, rs_last_rental)

pal <- colorFactor(c("navy", "red"), domain = c("Return", "Rental"))

# Purple: popular for both return and rental
# Red: only for rental
# Blue: only for return
# Size: total frequency within the quarter

# for Q2 of 2015
leaflet(Q2Y15) %>%
  addTiles() %>%
  addProviderTiles('CartoDB.Positron') %>%
  setView(lng=-75.163763, lat=39.952568, zoom = 13) %>%
  addCircleMarkers(
    radius = ~sqrt(count)/10,
    color = ~pal(type),
    stroke = FALSE, fillOpacity = 0.5, 
    popup = ~name
  )

# for Q4 of 2016
leaflet(Q4Y16) %>%
  addTiles() %>%
  addProviderTiles('CartoDB.Positron') %>%
  setView(lng=-75.163763, lat=39.952568, zoom = 13) %>%
  addCircleMarkers(
    radius = ~sqrt(count)/10,
    color = ~pal(type),
    stroke = FALSE, fillOpacity = 0.5, 
    popup = ~name
  )
```

<br> In the above two interactive maps, the color purple represents popular rental and return stations, red for rental, and blue for return. The size of the dot denotes the total frequency of rides within the quarter (top: Q2 of 2015, bottom: bottom: Q4 of 2016). Crudely in line with our hypothesis, most of the most popular stations (by dot size) are concentrated in the epicenter of Philadelphia, which is roughly between Rittenhouse Square and Chinatown. We also see that PMOA is off in the distance to the northeast, but was a popular rental and return station in Q2 of 2015. This also corresponds to our theory that summer made this a popular hub for bike users. In Q4 of 2016, the shift in clustering of popular stations revolves around the two shores of Schuylkill River. This may corroborate our observation that people use it as a means to cross the river for routine commutes (e.g. between University City/Powelton Village and Rittenhouse Square) that are out of walking range but close enough for a bike ride. This is purely speculative but makes for an interesting idea to further investigate. We will touch on this a bit more in the next section.

#### Daily traffic

We put the theory of using Indego as a form of commute to the test, using daily traffic as a proxy for measurements. We picked Q2 of 2016, which is the quarter with the second most bike rides. We ideally would have liked to chosen Q3 of 2016 (most recent summer quarter), which is the most popular quarter, but this time period overlaps with summer vacation for many students. Since we did not want to ignore this particular demographic when analyzing the use of bike trips as a form of commute, we thus settled on Q2 of 2016. We considered average hourly uses of Indego bikes per day for every day of Q2 of 2016. We then plotted this on a 24-hour time-scale, which is shown below. As illustrated by the plot, there are drastic differences between weekday and weekend usage across any standard 24-hour window. On weekdays, peak periods of bike trips were around 8:00 AM and 5:00 PM, which correspond to the time frames that most individuals commute respectively to and from work and school. On weekends, however, peak usage occured during the afternoon. Moreover, more bike trips were observed on weekdays for almost any given time interval than that of weekends. Thus, this may potentially demonstrate that Indego has become a means for everyday commute in Philadelphia. 

```{r}
# 2. Peak Periods (group by hour, weekdays/weekends, season)
# 2.1 Weekdays/weekends histogram of start (rental) hours for all stations in 2016Q2
library(lubridate)
Q2Y16 <- df[df$year==2016 & df$month %in% c("04","05","06"), ]
Q2Y16$day_type <- chron::is.weekend(Q2Y16$date)
Q2Y16$day_type <- ifelse(Q2Y16$day_type == 1, "Weekends", "Weekdays")
n_wknds <- length(unique(Q2Y16[Q2Y16$day_type=="Weekends",14]))
n_wkdys <- length(unique(Q2Y16[Q2Y16$day_type=="Weekdays",14]))
wknds_peak <- Q2Y16[Q2Y16$day_type=="Weekends",] %>% group_by(s_hour,day_type) %>% summarize(rental_freq = n()/n_wknds)
wkdys_peak <- Q2Y16[Q2Y16$day_type=="Weekdays",] %>% group_by(s_hour,day_type) %>% summarize(rental_freq = n()/n_wkdys)
peak <- rbind(wknds_peak, wkdys_peak)
rm(wknds_peak, wkdys_peak)
peak$s_hour <- as.numeric(as.character(peak$s_hour))
foo1 <- c(0,4,8,12,16,20)
foo2 <- c("12am","4am","8am","12pm","4pm","8pm")
```


```{r fig.width=3, fig.height=3}
ggplot(peak, aes(s_hour, rental_freq)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Peak Rental Hours 2016Q2",
       subtitle="Average Daily Rental Frequency by Hour") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)
```

We now extend our analysis to all quarterly data available, which is shown in the plot immediately below. Much of the same trends from Q2 of 2016 can be generalized to other quarters. Apart from a few data points, the number of bike trips on weekdays are almost always larger than that of weekends. However, we also observe the aforementioned seasonality of bike trips, in which more bike trips are observed in the summer than in the winter months. This is especially true for trips on weekend (e.g. usage on weekends in Sept of 2015 is by far the largest). Therefore, use of Indego as leisure during summer weekends may be the causal variable of the peaks we observe. 

```{r}
# 2.2 Month aggregation of results
quarter_aggr <- df
#quarter_aggr$quarter <- (as.numeric(as.character(quarter_aggr$month)) - 1) %/% 3 + 1
quarter_aggr$day_type <- chron::is.weekend(quarter_aggr$date)
quarter_aggr$day_type <- ifelse(quarter_aggr$day_type == 1, "Weekends", "Weekdays")
quarter_aggr <- quarter_aggr %>% select(year, month, date, day_type)
quarter_aggr <- quarter_aggr %>% group_by(date, year, month, day_type) %>% summarise(count=n())
quarter_aggr <- quarter_aggr %>% group_by(year, month, day_type) %>% summarise(average=mean(count))
quarter_aggr$no <- (0:(nrow(quarter_aggr)-1)) %/% 2 + 1
foo1 <- seq(1,21,by=3)
foo2 <- c("Apr 2015","Jul 2015","Oct 2015", "Jan 2016", "Apr 2016","Jul 2016","Oct 2016")
```

```{r fig.width=3, fig.height=3}
ggplot(quarter_aggr, aes(no, average)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Monthly Rental Pattern",
       subtitle="Daily Average of Rental Frequencies") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2) +
  theme(axis.text.x = element_text(angle = 30, hjust = 1))
```


It is important to note that one of the more glaring deficiencies of our raw dataset is the lack of historical data; we only have data from April 2015 onwards. It thus makes it more difficult to draw conclusive remakrs regarding yearly or even seasonal patterns that require data over longer time periods. Nonetheless, several important trends can still be observed over the time period for which the dataset is available.

For instance, let us take a look at monthly bike trips as determined from the number of average trips per day, shown below. Such an aggregation, as opposed to taking the monthly average as is, was done to smooth out daily fluctuations that can be considered noise as we make our time scale more granular. We observe two main themes. First, there seems to be particular months in which bike trips are more and less frequent. Across any given calendar year, there are more bike rides through the summer, ultimately peaking in September. Bike trips decline thereafter, hitting low points in December/January. This is intuitive as good weather is correlated with how willing and practical bike rides function as a means of transportation. Thus, with abysmal weather conditions in the winter, people are probably less willing to use Indego bikes. Second, we observe Indego's popularity increase through time based on the number of bike trips in a given month as compared to the same month in a preceding year. In Sept 2016, usage was at an all-time high, with more than 2600 rides per day (cf. approximately 2250 rides in Sept 2016). In Dec 2016, bike rides were roughly a little higher than 1000 rides per day (cf. 600 in Dec 2015). It should be noted that confounding factors such as weather conditions in a given year (e.g. El Nino) and the insufficient historical data we have on-hand may somewhat render our results questionable. We, however, believe this is not the case since more ecofriendly means of transportation should theoretically become more popular due to growing environmental sentiments in the mass population.

```{r results="hide",fig.width=3, fig.height=3}
# Monthly Bike Trips
df_count_by_month <- df %>% group_by(year, month) %>% summarize(count = n())
df_average_by_month <- df %>% group_by(year, month, day) %>% summarize(count = n())
df_average_by_month <- df_average_by_month %>% group_by(year, month) %>% summarize(avg = mean(count))
df_count_by_month$date <- zoo::as.yearmon(paste(df_count_by_month$year, df_count_by_month$month), "%Y %m")
df_count_by_month$average <- df_average_by_month$avg
df_count_by_month$no <- 1:nrow(df_count_by_month)
df_count_by_month <- df_count_by_month[,-c(1,2)]
rm(df_average_by_month)
df_count_by_month <- tidyr::gather(df_count_by_month, type, value, c(1,3))

foo1 <- c(0, 500, 1000, 1500, 2000, 2500, 3000)
foo2 <- c(3, 6, 9, 12, 15, 18, 21)

ggplot() +
  geom_line(aes(no, value, color="blue"),
            df_count_by_month[df_count_by_month$type=="average",]) +
  geom_point(size = 1.1) +
  ggtitle("Monthly Indego Bike Trips\n(Average per day)") +
  xlab("Month") +
  ylab("Count") +
  scale_color_fivethirtyeight("cyl") +
  theme_fivethirtyeight() +
  theme(axis.text.x = element_text(angle = 30, hjust = 1),legend.position="none") +
  scale_y_continuous(limits = c(0, 3000), breaks = foo1) +
  scale_x_continuous(breaks = foo2, labels = df_count_by_month$date[foo2]) +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm"))
```

As an interesting tangent, we examine the most popular routes based on a daily average of trips betweeen all stations across available quarters. We see that the most popular routes frequented by most users are those between hubs (i.e. already popular stations). In the interactive map below, we plot the Euclidean distance of the top five most popular routes. For instance, one such route is that between University Station and 23rd and South (i.e. downtown). It is worth noting that most of these routes are either across the downtown area or across the Schuylkill River, which somewhat substantiates the assertion that popular routes are a means for commute.

```{r, warning=FALSE, message=FALSE}
######## Popular Routes ######## 
# Content
# Most Popular 20 Routes
# Single dots means rent from these stations and return to these stations as well
popular_route <- df %>% group_by(date, year, month, start_station_id, end_station_id) %>% summarise(count=n())
popular_route <- popular_route %>% group_by(start_station_id, end_station_id) %>% summarise(avg=mean(count)) %>% na.omit
popular_route <- popular_route[order(-popular_route$avg),][1:20,]
# Extract all points
loc <- c(popular_route$start_station_id, popular_route$end_station_id) %>% unique %>% as.integer
station_loc$id <- as.numeric(station_loc$id)
dot_loc <- data.frame(id=loc) %>% left_join(station_loc, by=c("id"="id")) %>% na.omit

l1 <- leaflet(dot_loc) %>%
  addTiles() %>%
  addProviderTiles('CartoDB.Positron') %>%
  setView(lng=-75.163763, lat=39.952568, zoom = 13) %>%
  addCircleMarkers(
    color = "navy",
    stroke = FALSE, fillOpacity = 0.5, 
    popup = ~name
  )
popu <- popular_route[popular_route$start_station_id!=popular_route$end_station_id,]
dot_loc2 <- dot_loc
colnames(popu)[1] <- "id"
popu$id <- as.numeric(popu$id)
popu <- popu %>% left_join(dot_loc2, by=c("id"="id")) %>% na.omit
colnames(popu) <- c("sid", "id", "avg", "s_lat", "s_long", "name", "date")
dot_loc2$id <- as.numeric(dot_loc2$id)
popu <- popu %>% left_join(dot_loc2, by=c("id"="id")) %>% na.omit
colnames(popu) <- c("sid", "id", "avg", "s_lat", "s_long", "s_name", "date", "e_lat", "e_long", "e_name", "date_e")
for(i in 1:nrow(popu)){
  l1 <- addPolylines(l1, lat = as.numeric(popu[i, c(4, 8)]), lng = as.numeric(popu[i, c(5, 9)]))
}
l1
```

Based on the above findings, we can say with a certain level of confidence that the volume of bike trips is an important indicator of Indego's utility rate over different cross-sections of time. However, it does little to tell us how (not how much) Indego is being utilized. With this in mind, we motivate our next portion of analysis.

Since the inherent purpose of bike share networks are to mediate relatively short commutes, we want to examine the length of each trip. We surmise that shorter trips (0-30 minutes) should be the ideal range since anything beyond that is simply exercise and other transportation methods become increasingly more appealing. We consequently explore features related to the length of bike rides next.

#### Trip durations

In this section, we try to elucidate the more salient patterns pertaining to trip durations. From the two histograms below, which illustrate the durations of all trips in Q2 and Q4 of 2016 across weekdays and weekends, we observe that most trips taken were under 15 minutes. These two particular quarters were chosen since they are the latest quarters that also offers a contrast in seasonality. During weekdays for both quarters, shorter commutes (< 15 minutes) were the overwhelming majority of trip durations. They make up more than 70% of the total count. The same is true for weekends, but to a lesser extent. This may once again be an indicator that bikes used during weekdays are for commute (e.g. to and from work and school) so they are inherently shorter, whereas bikes are used for more recreational purposes on weekends so their time of use is naturally longer. The latter is corroborated by the fact that during summer weekends, the frequency of longer trips outweighed that of shorter trips.

If we ignore any differentiation of trip durations between weekdays and weekends, we observe much of the same trends. For Q4 of 2016, the latest quarterly data available to us, the most prevalent duration of bike rides lie between 5 and 10 minutes. In fact, the overwhelming majority of trips are less than 30 minutes. On the contrary, we see that long trips are actually outliers in our dataset. Trips between two and four hours only make up 0.9% of all rides in Q4 of 2016, and trips exceeding 4 hours make up 0.7% of the total.

```{r}
# 1. General Pattern Visualization
# 1.1 Trip Duration Distributions (dimension of time)
# Weekdays/weekends in 2016Q2 and 2016Q4
library(ggthemes)
Q2Y16 <- df[df$year==2016 & df$month %in% c("04","05","06"), ]
Q2Y16$day_type <- chron::is.weekend(Q2Y16$date)
Q2Y16$day_type <- ifelse(Q2Y16$day_type == 1, "Weekends", "Weekdays")
Q4Y16 <- df[df$year==2016 & df$month %in% c("10","11","12"), ]
Q4Y16$day_type <- chron::is.weekend(Q4Y16$date)
Q4Y16$day_type <- ifelse(Q4Y16$day_type == 1, "Weekends", "Weekdays")
Q2Y16$duration_cut <- cut(as.numeric(Q2Y16$duration), c(0,5,10,15,20,30,60,120,240,1440))
Q4Y16$duration_cut <- cut(as.numeric(Q4Y16$duration), c(0,5,10,15,20,30,60,120,240,1440))
Q2Y16_dur <- Q2Y16 %>% group_by(day_type, duration_cut, date) %>% summarise(count=n())
Q2Y16_dur<- Q2Y16_dur %>% group_by(day_type, duration_cut) %>% summarise(average=mean(count))
Q2Y16_dur$Quarter <- "2016Q2"
Q4Y16_dur <- Q4Y16 %>% group_by(day_type, duration_cut, date) %>% summarise(count=n())
Q4Y16_dur<- Q4Y16_dur %>% group_by(day_type, duration_cut) %>% summarise(average=mean(count))
Q4Y16_dur$Quarter <- "2016Q4"
dur_cut <- rbind(Q2Y16_dur, Q4Y16_dur)
rm(Q2Y16_dur, Q4Y16_dur, Q2Y16, Q4Y16)
foo <- c("<5min", "5-10min", "10-15min", "15-20min", "20-30min", "30-60min", "1-2hr", "2-4hr", ">4hr")
```

```{r fig.width=5, fig.height=3}
ggplot(dur_cut, aes(duration_cut, average)) +
  geom_bar(aes(fill=Quarter), stat = 'identity', position = position_dodge()) +
  facet_grid(. ~ day_type, scales = "free") +
  labs(title="Daily Trip Duration Histogram",
       subtitle="For 2016Q2 and 2016Q4",
       xlab="") +
  scale_color_fivethirtyeight(" ") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_discrete(labels = foo) +
  theme(axis.text.x = element_text(angle = 30, hjust = 1)) + 
  scale_fill_manual(values = c("#0072B2", "#CC79A7"))
```

We next examine the median rental duration for each station, particularly for Q2 to Q4 of 2016. We chose this range since it was the only range of contiuous quarterly data that had complete station entries for trip durations. In the interactive map below, larger circles correspond to larger values with respect to trip durations. The color scheme remains the same as previously mentioned. Therefore, a larger red circle implies that a bike user rented from a particular station and rode their bike for a longer duration. We see that larger red circles correspond to stations in the outskirts of Philadelphia. Therefore, they either used the bike for a longer commute or they used for leisurely purposes. For instance, one of the larger red circles denotes PMOA, which we previously mentioned as a popular station during the summer and weekends. However, it is impossible to differentiate between the two based on our dataset alone. It is however intriguing in nature.

```{r}
# 1.2 Median Rental Duration for Each Station (2016 Q2-Q4) (dimension of space)
# choose 2016 Q2-Q4 since all stations have complete data in this period
station_ana2 <- df[df$year==2016 & df$month %in% c("04","05","06","07","08","09","10","11","12"), ]
median_stat <- station_ana2 %>% group_by(start_station_id) %>% summarise(med=median(duration))
# Plot in a map, dot size = median rental time
colnames(median_stat)[1] <- "id"
station_loc$id <- as.numeric(station_loc$id)
median_stat$id <- as.numeric(median_stat$id)
median_stat <- median_stat %>% left_join(station_loc, by=c("id"="id")) %>% na.omit
content <- paste("<b>", as.character(median_stat$name), "</b><br>",
                 "Median Rental Time: ", median_stat$med, "minutes", "<br>")
median_stat$chosen <- median_stat$id %in% c("3020", "3050", "3124")
median_stat$chosen <- ifelse(median_stat$chosen == 1, "Chosen", "Unchosen")
pal <- colorFactor(c("navy", "red"), domain = c("Chosen", "Unchosen"))
```

```{r, warning=FALSE, message=FALSE}
leaflet(median_stat) %>%
  addTiles() %>%
  addProviderTiles('CartoDB.Positron') %>%
  setView(lng=-75.163763, lat=39.952568, zoom = 13) %>%
  addCircleMarkers(
    radius = ~med/2,
    color = ~pal(chosen),
    stroke = FALSE, fillOpacity = 0.5, 
    popup = ~content
  )
```

<br>The dots colored in navy are those that we chose to further analyze, which we present here. They correspond to the following stations:

* 3020: University City Station
* 3050: 9th & Arch (downtown)
* 3124: Race Street Pier (away from university and downtown)

```{r fig.width=3, fig.height=3, warning=FALSE, message=FALSE}
# 2. Station-level rental time

# Select stations
# Three locations that could have totally different travel patterns:
# 3020: University City Station, global avg duration 17.8
# 3050: 9th & Arch, (downtown), global avg duration 22.4
# 3124: Race Street Pier, (away from university and downtown), global avg duration 43.4

# 2.1 Over hour of a day
station_ana <- df[df$year==2016 & df$month %in% c("04","05","06","07","08","09","10","11","12"), ]
foo <- station_ana %>% group_by(start_station_id) %>% summarise(average=mean(duration)) # select stattions
station_ana <- station_ana[station_ana$duration<240,] # !! Truncate duration at 4 hrs to remove outliers
station_ana <- station_ana[station_ana$start_station_id %in% c("3020", "3050", "3124"),]
station_ana$day_type <- chron::is.weekend(station_ana$date)
station_ana$day_type <- ifelse(station_ana$day_type == 1, "Weekends", "Weekdays")
foo <- station_ana %>% group_by(start_station_id) %>% summarise(count=n()) # check total trips
station_ana <- station_ana %>% group_by(start_station_id, day_type, s_hour) %>% summarise(avg_dur=mean(duration))
foo <- data.frame(start_station_id = rep(c("3020", "3050", "3124"),each=48),
                  day_type = rep(rep(c("Weekdays", "Weekends"),each=24),3),
                  s_hour = rep(0:23,6))
station_ana$s_hour <- as.numeric(as.character(station_ana$s_hour))
station_ana$start_station_id <- as.character(station_ana$start_station_id)
station_ana$day_type <- as.character(station_ana$day_type)
station_ana <- station_ana %>% right_join(foo, by=c("start_station_id"="start_station_id",
                                                    "day_type"="day_type",
                                                    "s_hour"="s_hour"))
station_ana[is.na(station_ana)] <- 0
foo1 <- c(0,4,8,12,16,20)
foo2 <- c("12am","4am","8am","12pm","4pm","8pm")
ggplot(station_ana[station_ana$start_station_id=="3020",], aes(s_hour, avg_dur)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Average Trip Duration 2016 Q2-Q4",
       subtitle="At (3020) University City Station, Total Trips: 11920") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)

ggplot(station_ana[station_ana$start_station_id=="3050",], aes(s_hour, avg_dur)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Average Trip Duration 2016 Q2-Q4",
       subtitle="At (3050) 9th & Arch, Total Trips: 6121") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)

ggplot(station_ana[station_ana$start_station_id=="3124",], aes(s_hour, avg_dur)) +
  geom_line(aes(color=day_type)) +
  geom_point(size = 1.1, aes(color=day_type)) +
  labs(title="Average Trip Duration 2016 Q2-Q4",
       subtitle="At (3124) Race Street Pier, Total Trips: 3729") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)
```


In the first plot (University City Station), we consdier the station close to a university. We see that on average, longer trips occur on the weekend. In fact, none of the trips on weekdays exceed 25 minutes. We also observe that in both cases, longer trips tend to occur during the day, whereas shorter trips occur in the early morning and towards night time. In the second plot delineating the downtown station (9th & Arch), we see much of the same trend as that of the previous case. However, the discrepancy between trip durations across a 24-hour window on the weekends and the weekdays is much more pronounced. In our last plot above (Race Street Pier), we consider a station far from a university or downtown, which we can consider to be "more rural." Here, trips on weekdays and weekends mirror one another, showing they are closely related. Moreover, trips from this more rural station are all much longer, most exceeding 20 minutes irrespective of time intervals during the week. Therefore, it would seem that different stations based on their geographical location serve a different purpose for bike riders. Stations in more densely populated areas tend to favor shorter rides, whereas longer rides are more prevalent in more remote locations. It is important to note that the magnitude of number of trips may introduce some sort of counfounder on our insights (e.g. data imbalance), but we ignore this here, since it is difficult to address this issue given the limited historical data we have. This is a problem that would be solved naturally as more data becomes available. 

```{r fig.width=3, fig.height=3}
# 2.2 Over month of a year
station_ana2 <- df[df$year==2016 & df$month %in% c("04","05","06","07","08","09","10","11","12"), ]
station_ana2 <- station_ana2[station_ana2$start_station_id %in% c("3020", "3050", "3124"),]
station_ana2 <- station_ana2[station_ana2$duration<240,] # !! Truncate duration at 4 hrs to remove outliers
station_ana2 <- station_ana2 %>% group_by(start_station_id, month) %>% summarise(med=median(duration))
colnames(station_ana2)[1] <- "id"
station_ana2$id <- as.numeric(station_ana2$id)
station_ana2 <- station_ana2 %>% left_join(station_loc, by=c("id"="id"))
station_ana2$month <- as.numeric(as.character(station_ana2$month))
foo1 <- 4:12
foo2 <- c("Apr","May","Jun","Jul","Aug","Sept","Oct","Nov","Dec")
ggplot(station_ana2, aes(month, med)) +
  geom_line(aes(color=name)) +
  geom_point(size = 1.1, aes(color=name)) +
  labs(title="Median of Trip Duration",
       subtitle="Over Months (2016 Q2-Q4)") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2)
```

We realize it may be difficult to examine the trends of all three plots concurrently so we thus plot the median trip duration of all three stations in one single plot, which is shown above. It is interesting to note that over time, shorter rides become the prevalent theme. This is true for all three stations. However, the more rural station is still characterized by longer trips relative to the other two stations that are close to a university and city center.

Thus far, we have examined how and how much Indigo is utilized. However, we still do not know how users select to utilize Indego. For instance, which methods of payment are more common? Is there a change in preference over time? We seek to answer these questions in the next and final section of our analysis.

#### User profiles

In this section, we examine the attributes related to Indego bike users. Since the dataset provides only two dimensions regarding this aspect, we use the conventional mosaic plot to examine the proportions of different combinations of Indego users. These two features are passholder types (Indego30, IndegoFlex, and walkup) and type of trips (one-way, roundtrip).

Here is some more relevant information regarding the different [passholder types](https://www.rideindego.com/passes/):

* Indego30: $15/month, unlimited one hour trips
* IndegoFlex: $10/year base fee, and $4/hour
* Walk-up: $8/hour

```{r}
# 1.1 Mosaic plot of overall data
all_data_mosaic <- df %>% group_by(passholder_type, trip_route_category) %>% summarize(count = n())
colnames(all_data_mosaic) <- c("Pass", "Trip", "count")
```

```{r fig.width=3, fig.height=3}
library(vcd)
vcd::mosaic(xtabs(count ~ Trip + Pass, data = all_data_mosaic),
            direction = c("v", "h"),
            gp = gpar(fill = c("light green", "light blue", "light yellow")),
            spacing = spacing_highlighting,
            labeling = labeling_border(rot_labels = c(0,90,0,0)))
```

For all available quarters, we can see that the majority of users drastically favor one-way over round trips, and Indego30 passes over walkup payment plans. IndegoFlex is the least popular option regardless of route type. In fact, the combination of one-way trips with Indego30 is overwhelming characteristic, accounting for >50% of all users, followed by one-way with walk-ups. However, this is a extremely rough representation of user profiles since we ignore how user profiles vary over time. We examine this next.

Let us examine the most popular option (Indego30 x one-way), as shown below. We see that this option has become more popular at a steady rate, in which more than 75% of users choose this option in the lastest quarter.

```{r}
# 1.2 Type of passholders, frequency of round-trips (for month)
# for most popular type of trip route and pass type

# 2. Tendency of Moving to Most Popular Category
# Plot of the percentage of (passholder_type, trip_route_category) over quarters
# For most popular category
perc_category <- df %>% group_by(passholder_type, trip_route_category, year, month) %>% summarize(count = n())
perc_category$quarter <- (as.numeric(as.character(perc_category$month)) - 1) %/% 3 + 1
perc_category <- perc_category %>% group_by(year, quarter, passholder_type, trip_route_category) %>% summarise(count_qt=sum(count)) 
perc_category$type <- paste(perc_category$trip_route_category, "x", perc_category$passholder_type)
perc_category <- perc_category %>% group_by(year, quarter) %>% mutate(total=sum(count_qt))
perc_category$perc <- round(100*(perc_category$count_qt / perc_category$total),2)
perc_category <- perc_category[,-c(3:5,7)]
perc_category$no <- rep(1:7,each=6)
foo1 <- 1:7
foo2 <- c("2015Q2","2015Q3","2015Q4", "2016Q1", "2016Q2","2016Q3","2016Q4")
foo3 <- c("One Way x Indego30")
tmp1 <- perc_category[perc_category$type %in% foo3,]
```

```{r fig.width=3, fig.height=3}
ggplot(tmp1, aes(no, perc)) +
  geom_line(aes(colour=type)) +
  geom_point(aes(colour=type),size = 1.1) +
  labs(title="Percentage Change Over Quarters",
       subtitle="For Most Popular Category\nOne Way x Indego30") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2) +
  scale_y_continuous(limits = c(0, 100))
```

We see from the previous mosaic plot that round trips are not popular on average. We want to test whether there does indeed exist a decline in popularity over time. We see from the percentage change over quarters shown below that regardless of pass types, round trips make up an increasingly smaller fraction of the user demographic. We also postulate that the already unpopular combination of round trips and Indego Flex would make up an increasingly smaller proportion as well since they are more frequent travellers that would gradually move towards cheaper plans. This however is speculative and is contingent on a robust, large enough longitudinal dataset, which we do not have. Nontheless, this problem, albeit interesting, is beyond the scope of our analysis.

```{r}
# For roundtrips
foo4 <- c("Round Trip x IndegoFlex", "Round Trip x Indego30", "Round Trip x Walk-up")
tmp2 <- perc_category[perc_category$type %in% foo4,]
```

```{r fig.width=3, fig.height=3}
ggplot(tmp2, aes(no, perc)) +
  geom_line(aes(colour=type)) +
  geom_point(aes(colour=type),size = 1.1) +
  labs(title="Percentage Change Over Quarters",
       subtitle="Over Categories") +
  scale_color_fivethirtyeight("") +
  theme_fivethirtyeight() +
  theme(plot.margin = unit(c(0.5,0.5,0.5,0.5), "cm")) +
  scale_x_continuous(breaks = foo1, labels = foo2) +
  scale_y_continuous(limits = c(0, 7.15))

```

Based on the above analysis on user profiles, we see that as bike sharing becomes more popular over time in Philadelphia, more users opt for more cost-efficient payment plans, which is by far Indego30. The rise of one-way trips also conjecturally demonstrate that users are using Indego more frequently as a means of transportation over short distances or time periods. Obviously, we ignore many important factors related to user profiles. For instance, is there a preference for those renting from particular stations or during particular months? This is a natural extension of our analysis that we would like to further explore time permitting. However, we believe that our current set of analysis on user profiles presented here serves to provide an overarching representation of user profiles.

## Conclusion

In this project, we examined data related to Philadelphia's bike share program, Indego. We examined all available quarterly data (Q2 of 2015 to Q4 of 2016) to identify noteworthy patterns and trends related to the various ways that Indego is utilized, mainly pertaining to station usage, daily traffic, trip durations, and user profiles. We found that popular rental and return stations are concentrated around locations that are densely populated (e.g. in close proximity to city centers or university campuses). We also note that seasonality affects the usage rate of certain Indego stations. Using daily traffic as a proxy to determine whether or not Indego is used as a means of commute, we also found an increase in frequency of bike trips on weekdays that conveniently overlaps with time periods when most people commute to and from work or school. Moreover, we observe a higher volume of bike trips during the weekday, all of which conjecturally demostrate the vercaity of our aforementioned claim. 

We yet again observe trends of seasonality in daily traffic, in which volume during the summer far exceeds that of winter months. We also examined trip durations, and found that shorter trips (0-15 minutes) are the norm, but this is less evident for weekends than weekdays. Finally, we evaluated the proportions of Indego's user profiles and found that most users prefer monthly payments and one-way trips. We also found that over time, more individuals are opting for yearly memeberships, which may speak to the fact that Indego is becoming a more popular means of short-range commute in Philadelphia. 

It is worth mentioning that our analyses is extremely broad in nature. We would have liked to further explore certain trends and interaction effects. For instance, how does certain user profiles and stations correlate with each other? Are there any neighborhoods that have extremely high volumes of traffic? How does this vary monthly, quarterly, or even yearly? We would have liked to compared results across similar cities, and how they compare and constrast (e.g. with NYC's citibike). Time permitting, these are the more imminent avenues we would have liked to explore. 

Another caveat that should be brought to light is that the lack of historical data also makes it difficult to draw robust conclusions on year-to-year variability or even corroborate the significance of seasonal fluctuations. This problem obviously becomes less relevant as more quarterly data becomes available, but is a major obstacle in our current analysis for long-term time frames. 

Nonetheless, with the data available, we believe that the exploratory data analysis and visualizations that we provide here serve as a good first-order approximation of salient trends related to Indego and the ways it is being effectively utilized. We hope this serves as a pragmatic catalyst for other readers interested in the Indego dataset, or using the insights and techinques provided here to explore other similar datasets.

## Appendix: Shiny app

We also created a [Shiny application](https://github.com/masonsun/gr5702-proj) for real-time status updates for all active Indego stations. This is purely supplemental, and we provide snapshots here, and instructions to run our application in our Github repo linked above. You can follow the instructions on the website to run it on a local machine. 